# 计算机逻辑设计基础课程设计 小游戏 Flappy Mario 设计报告

Group: 费姚瑞 (3220103113)

仇国智(3220102181)

Date: 2023.12.22

## 一、背景介绍

Flappy bird 是一款 2D 横版闯关游戏,画面中一只小鸟始终向右飞行,玩家可以通过一直点击来让小鸟不断向上飞行,一旦停止点击小鸟就会开始下落。这个游戏要求玩家利用这一机制使小鸟保持适当的高度来通过不同高度的障碍物。

在本次课程设计中,我们小组将利用 Verilog HDL 语言和相关实验环境实现基于 Flappy bird 改编的 Flappy Mario 的运行和游玩。

Flappy\_Mario 将在 Flappy\_bird 的基础上加入双人游玩模式并增加一些细节内容。

### 二、设计说明

#### A. 设计开发环境

实验平台: Sword Kintex7

开发环境: Xilinx ISE、Xilinx Vivado

硬件描述语言: Verilog HDL

#### B. 输入输出交互选择

输入: Sword 平台上 clk 时钟信号与 SW[15]开关、PS2 键盘上 space 按钮及上下方向键

输出: Sword 平台上 VGA 显示器以及 4 位 7 段数码管

#### C. 游戏状态寄存器

Flappy Mario 游戏本质也是一个有限状态机,游戏中主要的状态寄存器如下:

- 1. status[1:0]: 01 为开始页面单人模式, 10 为开始页面双人模式, 00 为游戏中单人模式, 11 为游戏中双人模式。
- 2. score[15:0]: 储存游戏中获得的分数
- 3. pipe1~pipe3[31:0]: pipe\*储存上下一对水管的状态,一共三对

| 31~28 | 27~20 | 19~10  | 9~0    |
|-------|-------|--------|--------|
| 保留    | 水管间距  | 水管水平坐标 | 上面水管高度 |

4. Mario[15:0]: 储存角色状态

| 15        | 14~10 | 9~0  |
|-----------|-------|------|
| 状态1飞行,0下降 | 保留    | 角色高度 |

#### 5. coin[31:0]: 储存金币状态

| 31     | 30: 20 | 19: 10 | 9: 0  |
|--------|--------|--------|-------|
| 金币是否存在 | 保留     | 金币y坐标  | 金币x坐标 |

#### D. 总模块结构



#### E. 控制模块设计——Control Module

实现 Flappy Mario 这一小游戏的核心模块就是控制模块,即读入玩家的操作、将其反馈到游戏中并结算游戏状态。

Flappy Mario 这一游戏主要包括两种游玩模式,第一种为单人模式游玩, 其游玩规则与 Flappy bird 基本一致;第二种则为双人模式游玩,在单人模式 的基础上,双人模式将增加第二个玩家的操作,第二个玩家可以控制屏幕最 右侧(Mario 在屏幕左侧)的障碍物高度,从而为第一名玩家制造更加复杂 的障碍物,从而提高第一名玩家的游戏难度。

我们对这一小游戏的交互想法很简单,玩家的输入需要三个:一是控制, Mario 向上飞的 space 按钮,二是控制游戏是游玩状态还是初始状态的 SW[15]开关,三是用于第二名玩家控制障碍物高度的上下方向键。因此读入玩家的操作只需要三个一位的控制信号(up、rst、pipe\_up 和 pipe\_down)即可储存。

而对于游戏某一时刻的状态确定,我们需要以下信息:

- 1. clk: Sword 平台提供的 clk 时钟信号
- 2. fail: 游戏是否处于游玩状态,在拨动开始开关后游戏便开始游玩状态,倘若 Mario 撞到了障碍物上,则游戏失败,退出游玩状态。
- 3. rst: 控制游戏启动及重启,倘若 rst 为 1 则将游戏状态清 0,游戏回到 起始页面。
- 4. up: 记录玩家是否按下了控制 Mario 飞起的按钮, 若为 1 则 Mario 飞起
- 5. score: 记录本局游戏至此的分数, Mario 每成功穿过一个障碍物则加一分, Mario 每吃到一个金币则分数加二分, 初始为 0 分。
- 6. bird\_y: 记录当前状态下 Mario 的高度位置(以 Mario 左下角为基准点)以及 Mario 的行动状态(飞起与滑翔)
- 7. pipe: 记录当前障碍物(管子)的横坐标、高度(以管子左上角为基准点)以及这一建筑物供 Mario 穿过的间隙,屏幕中同时只出现三个障碍物。
- 8. coin: 记录当前金币的横坐标、纵坐标以及是否存在(若被 Mario 吃掉或从屏幕最左边离开则算作不存在)
- 9. pipe\_up 与 pipe\_down:记录玩家二是否按下控制障碍物高低的按钮, 若为1则使管子相应的上升或下降

#### 而对于控制模块的实现步骤如下:

- 1. 首先调用 clk\_20ms 模块,将 clk 周期放大为 20ms,并以 clk\_20ms 输出的时钟作为驱动时钟驱动以下步骤
- 2. 检测 rst 的值,倘若 rst 的值为 0,说明此时游戏尚未开始,游戏应当显示初始页面。
- 3. 在这一页面,玩家可通过按 PS2 键盘上的上下方向键,也就是pipe\_up、pipe\_down 信号来更改接下来的游玩模式(单人、双人),在代码中将用 status 二位变量记录, status=1 表示此时在初始页面选

- 择了单人模式, status=2 表示此时在初始页面选择了双人模式, status 默认为 1, 也就是默认为单人模式。
- 4. 同时在初始页面游戏处于准备状态,会在固定的三个位置生成随机高度的 pipe,也就是对 pipe\_x 赋初值,并对 pipe\_y 随机赋值,随机的范围为 pipe\_head~310-pipe\_head, pipe\_head 指的是水管一端的固定贴图,水管的长度至少要保证能使两端的端点贴图能完整显示,310 的值是 480(显示屏垂直分辨率)-170(水管间距最大值)得到的。同时需要随机生成上下水管的间距大小,也就是对 gap 随机赋值,范围为 120~170。
- 5. 在初始页面还要对金币的位置进行初始化,也就是对 coin[19:10]随 机赋值,范围为 20~440-coin\_length, coin\_length 为金币贴图的边长。 同时还需要对 Mario 的飞行状态、位置进行初始化,也就是对 bird\_y[15]赋值为 0、bird[14:0]赋值为 240(显示屏垂直分辨率的一半)。随机数的生成将调用 clk\_div 模块将 clk 计数,再将当前时间 的 clk\_div 根据指定范围取模从而获得对应的伪随机数。
- 6. 在初始页面选择了游玩模式后,可以拨动 SW[15]进入游戏状态,此时检测到 rst 的值为 1,则进入游戏状态。此时对 status 的值进行修改,倘若 status 的值为 1,则将其赋值为 0,表示单人模式正在游玩中,若 status 的值为 2,则将其赋值为 3,表示双人模式正在游玩中。
- 7. 游戏状态下,分别检测 up、fail 和 longpress 信号,longpress 信号用于检测是否长按,若 up 先前为 1,则 longpress 亦为 1 直到 up 变为=。
  - a) 当 up 为 1 (玩家按下 space)、fail 为 0 (游戏未结束) 且 longpress 为 0 (玩家刚刚按下 space),则将 bird\_flying 赋值为 10 (表示 Mario 将持续飞行 10 个时钟周期),将 bird\_y[15]赋值 为 1 (表示 Mario 正在飞行),并将 longpress 赋值为 1 (防止玩家长按 space)。
  - b) 当 bird\_flying 大于 0(Mario 还在惯性飞行)且 fail 为 0(游戏未结束),此时将 bird\_y 减少 bird\_flying 的值,相当于 Mario 向上飞了相当于 bird\_flying 大小的高度, 这样可以使 Mario 的飞

- 行速度先大后小,从而增强游戏手感;然后使 bird\_flying 的值减一。
- c) 当 bird\_flying 小于等于 0(Mario 不再飞行)或 fail 为 1(Mario 撞上障碍物,游戏结束)时,时 bird\_y 增加 4,相当于每个时 钟周期 Mario 下降 4 个像素点。
- d) 在此之外,对 up 进行判断,若 up 为 0,则令 longress 值变回 0
- 8. 游戏状态下,检测 fail 的值,若 fail 为 0,游戏未结束,此时 Mario 仍要向右飞行,也就是每个时钟周期将 pipe1\_x、pipe2\_x、pipe3\_x 以及 coin[9:0]减少 2,相当于 Mario 向右飞行了 2 个像素点。
- 9. 游戏状态下,检测 pipe1\_x、pipe2\_x、pipe3\_x 的值,若 pipe\_x<=2 (管子将要移出屏幕左端),则将 pipe\_x 重置为 640 (屏幕水平分辨率),并重新利用 clk\_div 对 pipe\_y 进行重新的随机初始化。
- 10. 双人模式游戏状态下,检测 pipe\_up 和 pipe\_down 信号,倘若 fail 值为 0(游戏未结束),且 status 值为 3(双人模式游玩中),则若 pipe\_up 信号为 1,则使屏幕最右侧的管子高度升高,若 pipe\_down 信号为 1,则使屏幕最右侧的管子高度降低。设置 cnt 变量,cnt 初始值为 3,用于记录当前屏幕最右侧的管子是几号管子,当 x 号管子从屏幕左侧离开在右侧出现时,cnt 将被赋值为 x,再利用 case 结构就可以根据 cnt 的值使 pipe\_up 和 pipe\_down 的效果作用于屏幕最右侧的管子。
- 11. 游戏状态下,检测 coin[31]和 coin[9:0]的值,倘若 coin[31]==0(金币被 Mario 吃掉)或是 coin[9:0]<=0(金币从屏幕左边移出),将 coin[9:0]的值重置为固定范围的随机值(范围在最右侧管子右边缘的右侧 0~20 个像素点,最右侧管子由 cnt 的值确定,避免出现金币和 pipe 重合的问题),coin[31]重置为 1,coin[19:10]的值重置为固定范围的随机值。
- 12. 游戏状态下,分别检测对于三根 pipe 的表达式((bird\_y[14:0]<= pipe\_y)||(bird\_y[14:0]+bird\_height>=pipe\_y+gap1))&&(bird\_x<=pipe\_x +pipe\_width)&&(bird\_x+bird\_width>=pipe\_x)的值,若有一个 pipe 对

应表达式值为 1,说明 Mario 撞到了障碍物,则游戏失败,fail 赋值为 1。

- 13. 游戏状态下,检测 fail 的值以及 pipe+pipe\_width 和 bird\_x 的值,若 fail=0 且 pipe+pipe\_width==bird\_x,说明此时 Mario 成功通过了一个 障碍物,则给 score 加一
- 14. 游戏状态下,在 fail==0 的前提下检测表达式 coin[31]&&((bird\_y[14:0]<=coin[19:10]+coin\_length)&&(bird\_y[14:0]+bird\_height>=coin[19:10])&&(bird\_x<=coin[9:0]+coin\_length)&&(bird\_x+bird\_width>=coin[9:0]))的值,若表达式值为 1,说明此时 Mario 成功吃到了金币,则给 score 加二
- 15. 最终将 gap、pipe\_x、pipe\_y 合并为 pipe 输出

为了保证游戏的合理性,此模块中需要添加 clk\_100ms 模块用于调整游戏速度,由于采用 PS2 键盘输入,不需要防抖模块。若要修改游戏运行速度则修改其中参数即可。



### F. 显示模块设计——Display Module

实现这一小游戏的另一重要模块便是显示模块,这一模块需要接收 control 模块运算得到的游戏状态并将其通过 VGA 可视化输出。

#### 模块输入:

- 1. clk: 系统时钟
- 2. status, pipe, Mario, score, coin: 游戏状态输入模块输出:

1. seg[6:0]:七段数码管输出

2. dp: 数码管小数点输出

3. AN[3:0]: 数码管共阳使能输出

4. rgb[11:0]: VGA12 位色彩输出

5. hs: VGA 水平同步信号

6. vs: VGA 垂直同步信号

模块内的主要组成部分如下:

- 1. 首先使用取余运算与截断除法将输入的十六进制分数转换为 BCD 码的十进制数据,然后调用 DispNum 模块将 BCD 码转换为数码管编码 (seg, dp, AN)输出。
- 2. 调用 VGA 接口模块 vga \_sync,模块输出当前对应需要输出的绝对 坐标和 VGA 同步信号 hs, vs。注意为了使得 VGA 输出帧率为 60 帧, 调用 clk 分频模块 clkdiv,输入周期为 25MHZ 的时钟。
- 3. 我们调用 Mario\_rom 等一系列用于储存图像信息的 rom IP 核,并根据当前绝对坐标和各个显示区域基准点坐标计算出相对坐标输入rom,获得 color Mario 等当前区域的对应像素值。
- 4. 显示图像大多是不规则平面图形,而矩形便于显示。为了便捷,所有图像区域均由不规则图形补全为矩形,设置透明色 12'h00f 填充补全区域像素点。
- 5. 为了实现金币转动的效果,设置寄存器 coin\_state[1:0]以储存金币转动角度的状态,每隔一定周期进行更新,通过 coin\_state 选择四幅金币旋转状态图中的一张进行显示. 与之相似的还有角色 Mario的显示,通过 Mario[15]来选择上升与下降图像。
- 6. 根据当前绝对位置坐标,和各个显示区域的位置的大小,我们使用 assign 语句和逻辑语句,来确定像素点是否位于标题文本、模式文本、高亮区域、管道头部、管道躯干、角色 Mario、金币图层的区域范围内,及该区域对应像素点是否为透明色,并使用 isMario 等一系列 wire 来表明结果。
- 7. 最后根据图层的覆盖关系和使能,使用三元运算符进行图层融合生

成最终的像素点。其中,为了便于根据 state[1:0]使能起始页面,我们先融合了标题文本、模式文本、高亮区域生成起始页面图层 color\_start。



#### G. VGA 接口模块——vga\_sync

VGA(Video Graphics Array)是一种模拟电脑视频接口标准,旨在连接显示器或电视设备,以便接收和显示来自电脑的图形信息。

1. VGA 使用模拟方式传输信号。其 15 针接口能分别传输红色、绿色、蓝色三种基色的模拟信号,以及相应的水平和垂直同步信号。 SWORD 实验板会根据 12 位颜色信号引脚的输入,在蓝绿红线上输出对应的电压强度。



2. VGA 通过行场同步信号来确定输出像素点的位置。行同步信号标志着视频扫描线的开始。每一行像素在显示时都会有一次行同步信号拉高,通过这个信号,显示器知道何时开始新的一行的扫描。场同步信号标志着一帧图像的开始。当显示器接收到场同步信号后,它知道一个新的画面开始,显示器即返回到屏幕的左上角,开始绘制新的一帧图像。行场同步信号时序如下,其中有效图像区间输出像素点。





VGA 可以显示不同分辨率的图像,不同分辨率及帧率对应参数如下:

| 显示模式         | 时钟     |     | 行同步信号时序(像素) |         |          |         |    | 场同步信号时序(行数) |    |    |     |          |         |       |           |
|--------------|--------|-----|-------------|---------|----------|---------|----|-------------|----|----|-----|----------|---------|-------|-----------|
|              | (MHz)  | 同步  | 后沿          | 左边<br>框 | 有效图<br>像 | 右边<br>框 | 前沿 | 行扫描周<br>期   | 同步 | 后沿 | 上边框 | 有效<br>图像 | 底边<br>框 | 前沿    | 场扫描居<br>期 |
| 640x480@60   | 25.175 | 96  | 40          | 8       | 640      | 8       | 8  | 800         | 2  | 25 | 8   | 480      | 8       | 2     | 525       |
| 640x480@75   | 31.5   | 64  | 120         | 0       | 640      | 0       | 16 | 840         | 3  | 16 | 0   | 480      | 0       | 1     | 500       |
| 800x600@60   | 40.0   | 128 | 88          | 0       | 800      | 0       | 40 | 1056        | 4  | 23 | 0   | 600      | 0       | 1     | 628       |
| 800x600@75   | 49.5   | 80  | 160         | 0       | 800      | 0       | 16 | 1056        | 3  | 21 | 0   | 600      | 0       | 1     | 625       |
| 1024x768@60  | 65     | 136 | 160         | 0       | 1024     | 0       | 24 | 1344        | 6  | 29 | 0   | 768      | 0       | 3     | 806       |
| 1024x768@75  | 78.8   | 176 | 176         | 0       | 1024     | 0       | 16 | 1312        | 3  | 28 | 0 / | [178E]   | (A) =   | TUATE |           |
| 1280x1024@60 | 108.0  | 112 | 248         | 0       | 1280     | 0       | 48 | 1688        | 3  | 38 | 0   | 1024     | 0 =     | 177   | T066      |

3. 最后我们进行 vga\_sync 模块的设计,根据输入的 25MHZ 时钟和 640\*480@60 行场同步信号的时序模式,输出 hs,vs 行场同步信号,并在有效图像区间根据时钟和输入颜色值,同步更新输出的绝对坐标(x,y)和12位颜色值。



### H. PS2 键盘输入模块——keyboard、kb\_decoder

SWORD 板与键盘是通过 USB 相连接,SWORD 板内部将其转换为 PS2 协议输入。

1. PS2 主要用于设计的两个引脚为输入时钟 kb\_clk,移位输入数据 data。主机读取 PS2 的 data 均在 kb\_clk 下降沿发生。如下为 data 读取时 11 位数据的时序结构。

| 0 | 起始位,总为0 |
|---|---------|

| 1~8 | 数据位,低位在前 |
|-----|----------|
| 9   | 校验位,奇校验  |
| 10  | 停止位,总为1  |



- 2. 在模块的输入端,我们可以看到有三个输入信号,包括系统时钟 clk,键盘时钟 kb\_clk 和键盘数据 data。在输出端,我们可以看到有两个输出信号,包括键盘码 keycode 和松开/按下信号 sign。注意为了达到同步赋值的效果,我们使用寄存器 kb\_clk\_old 存储键盘时钟的历史值,通过 kb\_clk\_old 检测 kb\_clk 的上升沿,并在 clk 的上升沿进行同步的赋值和更新。
- 3. 在模块内部,定义了一些寄存器,包括状态寄存器 state,临时键盘码寄存器 code\_temp,计数寄存器 cnt,确认寄存器 `confirm`和键盘时钟历史寄存器 kb\_clk\_old。state 用于表示当前的状态,包括等待(00)、输入(01)、确认(10)和结束(11)。code\_temp 用于暂存正在接收的键盘码。cnt 用于计数接收到的有效位数。confirm 用于存储奇偶校验位。
- 4. 在模块的主体部分,使用 always`语句在每个系统时钟上升沿更新寄存器和输出信号。首先,更新 kb\_clk\_old。然后,根据 state 的值执行不同的操作。在等待状态,如果检测到键盘时钟的下降沿,并且数据位为 0,表示开始接收一个新的键盘码,此时将 state 设置为输入状态,将 confirm 清零,将 sign 设置为 0 或 1, (sign 取决于前一位是否为 F0 即松开的前一半状态码)。在输入状态,如果检测到键盘时钟的下降沿,将 cnt 加 1,如果 cnt 的所有位都为 1,表示已经接收到 8 位数据,此时将 state 设置

为确认状态,将 cnt 清零,将 confirm 取反,将 code\_temp 左移并加上数据位。在确认状态,如果检测到键盘时钟的下降沿,并且确认位与数据位不同,表示接收到的键盘码正确,此时将 state 设置为结束状态,否则将 state 设置为等待状态,将 code\_temp 清零。在结束状态,将 state 设置为等待状态,将 keycode 设置为 code\_temp 的值。





#### I. ROM 寄存器——IP 核设计

ROM 全称为只读存储器(Read-Only Memory),是一种在断电后数据仍然不会丢失的存储器。ROM 有几种不同的类型,包括掩模 ROM(Mask ROM,被程序预先硬编码,不能被用户改写)、可编程 ROM(PROM,可以被用户写入数据,但只能写一次)、可擦写可编程 ROM(EPROM,可以通过紫外线进行擦写)、电可擦写可编程 ROM(EEPROM,可以通过电流进行擦写)等。在实际使用中,这些只读存储器主要用在一些需要长期储存数据,且数据不常更改的场合。基于此特点,ROM 在本实验中主要用于储存大量的图像数据的元件设计

本实验中采取 IP 核进行 ROM 设计。IP 核通常是指在使用 FPGA (现场可编程门阵列)设计时,预先设计好并可以重复利用的电子设计单元。

首先从网络上搜集游戏动画素材,转换合适大小后导出处理为.coe 文件格式。

然后选择 IP 核中的 ROM 模块。

| <b>→</b> N | New Source Wizard                     |                                       |      |  |  |  |  |  |  |  |  |
|------------|---------------------------------------|---------------------------------------|------|--|--|--|--|--|--|--|--|
| s          | Select IP                             |                                       |      |  |  |  |  |  |  |  |  |
|            | Create Coregen or Architecture Wizard | IP Core.                              |      |  |  |  |  |  |  |  |  |
|            | create coregen or recincectare means  |                                       |      |  |  |  |  |  |  |  |  |
|            |                                       |                                       |      |  |  |  |  |  |  |  |  |
|            | View by Function View by Name         |                                       |      |  |  |  |  |  |  |  |  |
|            | , , ,                                 |                                       |      |  |  |  |  |  |  |  |  |
|            | Name                                  | / Version AXI4 AXI4-Stream            |      |  |  |  |  |  |  |  |  |
|            | - 🏺 Block Memory Generator            | 7.3                                   |      |  |  |  |  |  |  |  |  |
|            | Distributed Memory Gener              | ator 7.2                              |      |  |  |  |  |  |  |  |  |
|            | □ D Video & Image Processing          |                                       |      |  |  |  |  |  |  |  |  |
|            | AXI Video Direct Memory Acc           | l.                                    | 7    |  |  |  |  |  |  |  |  |
|            | └ 🎖 Video Direct Memory Access        | 1.1                                   | 3    |  |  |  |  |  |  |  |  |
|            | <b>■</b> :::::                        | 1 1                                   | ╝    |  |  |  |  |  |  |  |  |
|            | Search IP Catalog: memory             | C <u>l</u> ear                        |      |  |  |  |  |  |  |  |  |
|            | All IP versions                       | Only IP compatible with chosen par    | t    |  |  |  |  |  |  |  |  |
|            |                                       | _ o, co <u></u> patione man enosemper | -    |  |  |  |  |  |  |  |  |
|            |                                       |                                       |      |  |  |  |  |  |  |  |  |
| 1          |                                       |                                       |      |  |  |  |  |  |  |  |  |
|            |                                       |                                       |      |  |  |  |  |  |  |  |  |
| Mo         | ore Info                              | < <u>B</u> ack <u>N</u> ext > Car     | icel |  |  |  |  |  |  |  |  |
|            |                                       |                                       |      |  |  |  |  |  |  |  |  |

调整 IP 核参数,途中 a 为位置输入引脚,spo 为数据输出引脚



导入.coe 文件。





#### J. 音频模块设计——buzzer\_controller

为增强游戏的游玩体验,课程设计中增加了音乐模块,实验台的 buzzer 按照设定好的频率和节奏发声即可放出音乐。

对于 buzzer 而言,放出音乐只需要两个信息,一是当前音符对应的 发声频率,二是当前音符需要持续多久,因此我们需要两个数组分别 是 note\_periods[]记录每个音符的振动周期和 note\_duration[]记录每个音符需要发声多久。

- 1. 首先需要对 note\_period[]和 note\_duration[]进行初始化,也就是保存要放的音乐的乐谱信息。
- 2. 接着只需要执行一个 always 循环,当如果当前音符的一个周期已经完成,反转 buzzer 的状态;如果当前音符已经播放完毕,切换到下一个音符;否则,增加计数器。

以上,便可实现一个音频 buzzer 模块。



### 三、调试过程分析

1. 首先是语法调试,语法调试阶段是非常重要的,这部分与其他现代编程语言的调试相似,错误定位通常是非常准确的。我会使用各种开发工具进行语法检查,包括但不限于代码编辑器的内建语法检查功能或者专门的

静态代码分析工具。如果我们正在处理一块大的代码,而且其中有很多错误,那么可能会一次性得到很多错误消息。这样的情况下,最好的策略是从上到下,一次解决一个错误,然后重新编译,再继续下一个。同时,让别人审查你的代码,或者对别人的代码进行审查,都是很好的学习和发现错误的方式。



. 然后,我们会运行项层代码的 RTL 综合,分析 RTL 综合电路,看综合结果是否有什么特别之处,这将确保设计功能是否一致。同时,我也会分析和浏览系统综合报告,以清楚地理解系统资源使用情况,检查各个模块的错误信息(包括警告信息),并一一排除。以下是我在分析综合结果时通常会执行的步骤:分析资源使用情况:我会检查设计对 FPGA 资源(如逻辑单元、内存块、I/O单元等)的使用,以确保设计不会超出器件的容量。如果资源利用率过高,我可能需要优化设计以提高资源利用率。检查时序报告:这个报告会告诉我设计是否满足时序限制。我会查看数据路径以找出可能存在的关键路径,并如果需要,进行时序优化。查看功耗报告:设计的耗电量是一项重要的指标,过高的功耗可能导致热问题,甚至影响设备的寿命。我会查看各个模块的功耗,特别是大单元块的功耗。检查编译器的警告和错误:编译器在综合过程中可能会返回一些警告或错误信息。我会仔细检查这些信息,以确保我们的设计能正确地实现预期的功能。验证设计满足功能需求:我会参照设计的功能描述,检查 RTL 综合结果是否满足这些要求。我也会通过运行模拟来进一步检

查设计的正确性。



- 3. 接下来是各个模块的调试过程,包括控制模块,显示模块,VGA 接口模块,PS/2 接口模块,蜂鸣器接口模块等等。我会编写良好的仿真激励代码,根据相应的仿真波形,认真分析,针对其中出现的变量定义错误、语句错误等错综复杂的问题进行逐一修改。具体细节见核心模块仿真部分。
- 4. 编写 ucf 文件对引脚进行定义,下载至 SWORD 实验板进行验证。在编写 UCF 文件时,需要参考实验板的手册,依据手册上的 FPGA 引脚分配表,将设计中的每一个输入输出信号逐一分配到对应的物理引脚上。注意输入输出信号的名称和信号类型必须跟设计中保持一致。接下来就是观察实验板上的表现,如角色马里奥上下移动情况,管子是否正常随机生成等实际现象。若实验板上的结果与预期一致,那么说明电路设计的功能已经成功实现,否则需要回过头去查找问题的原因,并进行代码的修正和优化,然后重复上述的验证过程。

### 四、核心模块仿真分析

#### 1. Control 模块仿真

初始状态下,rst、up、pipe\_up、pipe\_down 均为 0,然后按下 pipe\_up 按钮可以看到 status 从 1 变为 2,说明此时在标题页面从默认的单人模式变为了双人模式。然后在 120ns 拨动 rst 为 1, status 由 2 变为 3,说明此时以双人模

式开始游戏,接下来 bird\_y 不断减少,说明 Mario 在下落。而在 220ns 处连续按下三次 Up 后 bird\_y 增大了数个周期,说明 Mario 飞起。同时 240ns 时按下了 pipe\_up, 260ns 时按下了 pipe\_down,最右侧的管子 pipe3 高度相应地变大变小了。



在不操作后,从 2620ns 开始 coin、pipe 等值开始不变,说明此时 Mario 撞到障碍物,游戏停止。



在 3270ns 时将 rst 拨回 0,此时 status 又变回 1,说明此时游戏回到 初始页面。



#### 2. Display 模块仿真







seg 为 7 位数码管输出均为 0, AN 交错使能四位数码管, score 数码管输出正常。VGA 行场同步信号 vs、hs 符合时序逻辑图, rgb 输出信号经过分析符合所属部分区域的像素点。

### 五、游玩说明

### 游戏目标:

控制 Mario 飞行, 避开障碍物, 吃到金币, 获得更高分数

### 游戏游玩方式:

将游戏下载到实验板上后,将 SW[15]拨至 0 状态,此时游戏位于初始页面,此时玩家可以按动 PS2 键盘上的上下方向键在初始页面选择游玩模式(单人模式、双人模式),默认为单人模式。选择游玩模式后将 SW[15] 拨至 1 状态即可开始游戏:

单人模式: 只有一位玩家可以参与游戏, 按动 PS2 键盘的空格键可以使 Mario 飞起, 若不按则 Mario 会因重力下降, 玩家要靠这一机制使 Mario 保持合适高度通过障碍物、吃到金币, 每通过一个障碍物则分数加 1, 每吃到一个金币则分数加 2, 分数将被显示在实验台的 4 位数码管上。

**双人模式:** 有两位玩家参与游戏, 玩家 1 的操作同单人模式的玩家, 而玩家二可以按动 PS2 键盘上的上下方向键调整屏幕最右边管子的高度, 从而为玩家一制造更高难度的障碍, 分数规则同单人模式, 同样会显示在 4 位数码管上。

#### 注意:

- 1. 在游玩的任何时候都可以通过拨动 SW[15]来使游戏回到初始页面。
- 2. 游玩模式只有在初始界面可以选择,开始游玩后游玩模式将无法改变。

### 六、验证过程展示

初始页面切换模式高亮、显示均正常





Mario 过管得分正常



Mario 吃金币得分正常



Mario 撞到障碍物后游戏正常结束



# 七、成员及分工说明

3220103113 费姚瑞 50%: control、buzzer 模块设计,论文撰写 3220102181 仇国智 50%: display、VGA、PS2 接口模块设计,论文撰写